/* Contributed by Jeff Laing
 * 20080414
 */

#ifndef _PACKED_FONT_BASE_
#define _PACKED_FONT_BASE_

#include <nds.h>
#include "fontbase.h"

namespace WoopsiUI {

	/**
	 * PackedFont is a base class defining a font whose data is packed into a
	 * more efficient data format.
	 */
	class PackedFontBase : public FontBase
	{
	public:
		/**
		 * Constructor.
		 * @param first Ascii index of first character in glphyDdata.
		 * @param last Ascii index of last character in glyphData.
		 * @param glyphData Packed array representing font.
		 * @param glyphOffset Offset into glyphData[] of character[i].
		 * @param glyphWidth Pixel width of character[i].
		 * @param height The height of the font.
		 * @param spWidth The width of a space character.
		 * @param fontTop The height of the font minus the blank spaces below
		 * 'a'.
		 * @param fixedWidth Character width (fixed), or 0 for proportional.
		 */
		PackedFontBase(
			u8 first, u8 last,
			const u16 *glyphData,
			const u16 *glyphOffset,
			const u8 *glyphWidth,
			const u8 height,
			const u8 spWidth,
			const u8 fontTop,
			const u8 fixedWidth = 0)
			:
			 _height(height),
			 _first(first), _last(last),
			 _glyphData(glyphData), _glyphOffset(glyphOffset), _glyphWidth(glyphWidth),
			 _fontWidth(0), _spWidth(spWidth),
			 _fontTop(fontTop), _widMax(fixedWidth) { };

		/**
		 * Makes this font fixed-width, though doesn't allow the spacing to be
		 * less than the font's known maximum character width.  If you want
		 * fixed spacing but don't know the required size, pass 1.
		 * @param fontWidth Font width to use (0 for proportional, 1 for
		 * wide-enough).
		 */
		inline void setFontWidth(u8 fontWidth) { _fontWidth = (fontWidth && (fontWidth < _widMax)) ? _widMax : fontWidth; };

		/**
		 * Draw an individual character of the font to the specified bitmap.
         * @param bitmap The bitmap to draw to.
         * @param letter The character to output.
         * @param colour The colour to draw with.  If this is 0 the font's
         * default colour will be used.
         * @param x The x co-ordinate of the text.
         * @param y The y co-ordinate of the text.
         * @param clipX1 The left edge of the clipping rectangle.
         * @param clipY1 The top edge of the clipping rectangle.
         * @param clipX2 The right edge of the clipping rectangle.
         * @param clipY2 The bottom edge of the clipping rectangle.
         * @return The x co-ordinate for the next character to be drawn.
		 */
		virtual s16 drawChar(
			MutableBitmapBase* bitmap,
			u32 letter,
			u16 colour,
			s16 x, s16 y,
			u16 clipX1, u16 clipY1, u16 clipX2, u16 clipY2);
		
		/**
		  * Draw an individual character of the font to the specified bitmap on a
         * baseline.
         * @param bitmap The bitmap to draw to.
         * @param letter The character to output.
         * @param colour The colour to draw with.  If this is 0 the font's
         * default colour will be used.
         * @param x The x co-ordinate of the text.
         * @param y The y co-ordinate of the text.
         * @param clipX1 The left edge of the clipping rectangle.
         * @param clipY1 The top edge of the clipping rectangle.
         * @param clipX2 The right edge of the clipping rectangle.
         * @param clipY2 The bottom edge of the clipping rectangle.
         * @return The x co-ordinate for the next character to be drawn.
		 */
		virtual s16 drawBaselineChar(
			MutableBitmapBase* bitmap,
			u32 letter,
			u16 colour,
			s16 x, s16 y,
			u16 clipX1, u16 clipY1, u16 clipX2, u16 clipY2) {
		        return drawChar(bitmap, letter, colour, x, y - getCharTop(y), clipX1, clipY1, clipX2, clipY2);
			};			
		
		/**
		 * Get the width of a string in pixels when drawn with this font.
		 * @param text The string to check.
		 * @return The width of the string in pixels.
		 */
		virtual u16 getStringWidth(const WoopsiString& text) const;

		/**
		 * Get the width of a portion of a string in pixels when drawn with this
		 * font.  Useful if you want to determine the length of a string without
		 * a terminator, or the length of a section of a string.
		 * @param text The string to check.
		 * @param startIndex The start point of the substring within the string.
		 * @param length The length of the substring in chars.
		 * @return The width of the substring in pixels.
		 */
		virtual u16 getStringWidth(const WoopsiString& text, s32 startIndex, s32 length) const;

		/**
		 * Get the width of an individual character.
		 * @param letter The character to get the width of.
		 * @return The width of the character in pixels.
		 */
		virtual u8 getCharWidth(u32 letter) const;

		/**
		 * Get the top of an individual character (constant for a packedfont).
		 * @param letter The character to get the width of.
		 * @return The width of the character in pixels.
		 */
		inline virtual s8 getCharTop(u32 letter) const {return _fontTop;};

		/**
		 * Text uses this to decide whether a given character is blank or not
		 */
		virtual bool isCharBlank(const u32 letter) const;

		/**
		* Gets the height of the font.
		* @return The height of the font.
		*/
		virtual u8 getHeight() const { return _height; };
		
		/**
		* Get the height of an individual character.
		* @param letter The letter to get the height of.
		* @return The height of the character in pixels.
		*/
		virtual u8 getCharHeight(u32 letter) const { return _height; };
		
		/**
		 * Render an individual character of the font to the specified bitmap.
		 * @param pixelData The font-specific pixel data
		 * @param pixelsPerRow The number of pixels to render per row (for this
		 * character).
		 * @param bitmap The bitmap to draw to.
		 * @param colour The colour to draw with. If this is 0 the font's
		 * default colour will be used.
		 * @param x The x co-ordinate of the text.
		 * @param y The y co-ordinate of the text.
		 * @param clipX1 The left edge of the clipping rectangle.
		 * @param clipY1 The top edge of the clipping rectangle.
		 * @param clipX2 The right edge of the clipping rectangle.
		 * @param clipY2 The bottom edge of the clipping rectangle.
		 */
		virtual void renderChar(
			const u16* pixelData, u16 pixelsPerRow,
			MutableBitmapBase* bitmap,
			u16 colour,
			s16 x, s16 y,
			u16 clipX1, u16 clipY1, u16 clipX2, u16 clipY2) = 0;

	protected:
		u8 _height; 				/**< The height of the font. */
		u8 _first; 					/**< The first letter that the font contains. */
		u8 _last; 					/**< The last letter that the font contains. */
		const u16 *_glyphData; 		/**< All data for each glyph. */
		const u16 *_glyphOffset; 	/**< Locations of each character in _glyphData. */
		const u8 *_glyphWidth; 		/**< Width in pixels of each glyph in _glyphData. */
		u8 _fontWidth; 				/**< Width of the font, or 0 for proportional. */
		u8 _spWidth; 				/**< Width of a blank space. */
		u8 _fontTop;				/**< Constant Top of the packed font. */
		u8 _widMax; 				/**< The maximum width of a character in the font. */
	};
}

#endif
